package com.heima.media.service.impl;

import com.alibaba.fastjson.JSON;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.heima.common.dto.PageResponseResult;
import com.heima.common.dto.ResponseResult;
import com.heima.common.dto.User;
import com.heima.common.enums.AppHttpCodeEnum;
import com.heima.common.util.MediaThreadLocalUtil;
import com.heima.media.dto.*;
import com.heima.media.entity.WmNews;
import com.heima.media.entity.WmNewsMaterial;
import com.heima.media.entity.WmUser;
import com.heima.media.mapper.WmNewsMapper;
import com.heima.media.service.IAuditService;
import com.heima.media.service.IWmNewsMaterialService;
import com.heima.media.service.IWmNewsService;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.heima.media.service.IWmUserService;
import com.heima.media.vo.WmNewsVo;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Lazy;
import org.springframework.kafka.core.KafkaTemplate;
import org.springframework.stereotype.Service;
import org.springframework.util.StringUtils;

import java.util.ArrayList;
import java.util.Date;
import java.util.List;

/**
 * <p>
 * 自媒体图文内容信息表 服务实现类
 * </p>
 *
 * @author dxx
 * @since 2022-09-20
 */
@Service
public class WmNewsServiceImpl extends ServiceImpl<WmNewsMapper, WmNews> implements IWmNewsService {

    @Value("${topic.enableTopic}")
    private String enableTopic;

    @Autowired
    private KafkaTemplate<String, String> kafkaTemplate;

    @Autowired
    private IWmUserService userService;

    @Autowired
    IWmNewsMaterialService newsMaterialService;

    @Autowired
    @Lazy    // 解决循环依赖问题
    private IAuditService auditService;

    @Override
    public ResponseResult listByCondition(WmNewsPageDto dto) {

        // 从本地线程获取用户信息
        User user = MediaThreadLocalUtil.get();
        // 如果用户为空,提示需要登录
        if (user == null) {
            return ResponseResult.errorResult(AppHttpCodeEnum.NEED_LOGIN);
        }
        // 分页查询文章列表
        IPage<WmNews> page = new Page<>(dto.getPage(), dto.getSize());
        LambdaQueryWrapper<WmNews> query = new LambdaQueryWrapper<>();
        // 状态不为空,添加状态查询条件
        if (dto.getStatus() != null) {
            query.eq(WmNews::getStatus, dto.getStatus());
        }
        // 关键字不为空,添加关键字根据文章标题模糊查询
        if (!StringUtils.isEmpty(dto.getKeyword())) {
            query.like(WmNews::getTitle, dto.getKeyword());
        }
        // 频道id不为空,添加频道查询条件
        if (dto.getChannelId() != null) {
            query.eq(WmNews::getChannelId, dto.getChannelId());
        }
        // 开始日期不为空,添加发布时间查询条件
        if (dto.getBeginPubDate() != null) {
            query.ge(WmNews::getPublishTime, dto.getBeginPubDate());
        }
        // 结束日期不为空,添加发布时间查询条件
        if (dto.getEndPubDate() != null) {
            query.lt(WmNews::getPublishTime, dto.getEndPubDate());
        }
        // 添加当前登录用户的id查询
        query.eq(WmNews::getUserId, user.getUserId());
        // 根据文章的发布时间倒序排列
        query.orderByDesc(WmNews::getPublishTime);
        IPage<WmNews> iPage = this.page(page, query);
        PageResponseResult result = new PageResponseResult(dto.getPage(), dto.getSize(), iPage.getTotal(), iPage.getRecords());
        return result;
    }

    @Override
    public ResponseResult submit(WmNewsDto dto) {

        // 从本地线程中获取用户id
        User user = MediaThreadLocalUtil.get();
        if (user == null) {
            return ResponseResult.errorResult(AppHttpCodeEnum.NEED_LOGIN);
        }
        // 构建文章对象
        WmNews wmNews = new WmNews();
        BeanUtils.copyProperties(dto, wmNews);
        wmNews.setUserId(user.getUserId());
        wmNews.setCreatedTime(new Date());
        wmNews.setSubmitedTime(new Date());
        wmNews.setEnable(true);
        // 将封面图片转换成json保存
        String jsonString = JSON.toJSONString(dto.getImages());
        wmNews.setImages(jsonString);
        // 从内容中提取图片
        List<ImageDto> contentImage = getImageFromContent(dto.getContent());
        // 如果前端传递的类型是自动,也就是type = -1 ,需要根据内容图片做对应的处理
        if (dto.getType() == -1) {
            // 根据内容图片的数量来判断
            if (contentImage.size() <= 0) {
                // 内容图片数量=0,修改 type 为0 ,无图模式
                wmNews.setType(0);
            } else if (contentImage.size() < 3) {
                // 内容图片数量<3,修改 type 为1 ,单图模式,并且取第一张内容图片作为封面图片
                wmNews.setType(1);
                wmNews.setImages(JSON.toJSONString(contentImage.subList(0, 1)));
            } else {
                // 内容图片数量>=3,修改 type 为3 ,多图模式,并且取前三张内容图片作为封面图片
                wmNews.setType(3);
                wmNews.setImages(JSON.toJSONString(contentImage.subList(0, 3)));
            }
        }

        // 判断是否包含文章id,如果包含文章id,先删除关联关系,然后走的保存的逻辑
        if (dto.getId() != null) {
            deleteRelation(dto.getId());
        }

        // 保存文章
        // this.save(wmNews);
        this.saveOrUpdate(wmNews);

        // 如果状态是提交审核,需要保存文章和素材的关联关系,包含内容关联和封面关联
        if (dto.getStatus() == 1) {
            // 保存封面图片和文章的关联关系
            List<ImageDto> coverImages = JSON.parseArray(wmNews.getImages(), ImageDto.class);
            // saveRelation(wmNews, dto.getImages(), 1);
            saveRelation(wmNews, coverImages, 1);   // 自动类型前端传递的dto.getImages为空
            // 保存内容图片和文章的关联关系
            saveRelation(wmNews, contentImage, 0);

            // 开启异步审核
            System.out.println(Thread.currentThread().getName() + "调用审核服务...");
            auditService.audit(wmNews);
        }
        return ResponseResult.okResult();
    }

    private void saveRelation(WmNews wmNews, List<ImageDto> images, int type) {
        List<WmNewsMaterial> list = new ArrayList<>();
        int ord = 0;

        // 遍历图片
        for (ImageDto image : images) {
            WmNewsMaterial newsMaterial = new WmNewsMaterial();
            newsMaterial.setMaterialId(image.getId());
            newsMaterial.setNewsId(wmNews.getId());
            newsMaterial.setType(type);
            newsMaterial.setOrd(ord);
            list.add(newsMaterial);
            ord++;
        }
        // 批量保存关联关系
        newsMaterialService.saveBatch(list);
    }

    private void deleteRelation(Integer newsId) {
        LambdaQueryWrapper<WmNewsMaterial> query = new LambdaQueryWrapper<>();
        query.eq(WmNewsMaterial::getNewsId, newsId);
        newsMaterialService.remove(query);
    }

    private List<ImageDto> getImageFromContent(String content) {
        List<ImageDto> contentImage = new ArrayList<>();
        // 反序列化json
        List<ContentDto> dtoList = JSON.parseArray(content, ContentDto.class);
        for (ContentDto dto : dtoList) {
            if (dto.getType().equals("image")) {
                ImageDto imageDto = new ImageDto();
                imageDto.setId(dto.getId());
                imageDto.setUrl(dto.getValue());
                contentImage.add(imageDto);
            }
        }
        return contentImage;
    }


    @Override
    public ResponseResult enable(WmNewsDto dto) {

        // 根据id查询文章
        WmNews wmNews = this.getById(dto.getId());
        // 判断文章是否存在
        if (wmNews == null) {
            return ResponseResult.errorResult(AppHttpCodeEnum.DATA_NOT_EXIST);
        }
        // 判断文章是否发布
        if (wmNews.getStatus() == 9) {
            // 修改文章,修改上下架状态
            wmNews.setEnable(dto.getEnable() == 1 ? true : false);
            this.updateById(wmNews);
            // 同步状态到Article端  发送消息
            String key = "" + wmNews.getArticleId();
            String value = dto.getEnable() == 1 ? "true" : "false";
            kafkaTemplate.send(enableTopic, key, value);
        }

        return ResponseResult.okResult();
    }


    @Override
    public ResponseResult findPageByName(NewsAuthDto dto) {
        // 构建分页条件
        IPage<WmNews> page = new Page<>(dto.getPage(), dto.getSize());
        // 构建查询条件
        LambdaQueryWrapper<WmNews> query = new LambdaQueryWrapper<>();
        // 根据标题模糊查询
        if (!StringUtils.isEmpty(dto.getTitle())) {
            query.like(WmNews::getTitle, dto.getTitle());
        }
        // 根据状态查询
        if (dto.getStatus() != null) {
            query.eq(WmNews::getStatus, dto.getStatus());
        }
        // 根据发布时间倒序排列
        query.orderByDesc(WmNews::getPublishTime);
        // 分页查询
        IPage<WmNews> iPage = this.page(page, query);
        List<WmNews> newsList = iPage.getRecords();

        // 构建vo对象
        List<WmNewsVo> vos = new ArrayList<>();
        for (WmNews wmNews : newsList) {
            WmNewsVo vo = new WmNewsVo();
            // 属性赋值
            BeanUtils.copyProperties(wmNews, vo);
            // 查询作者信息
            // todo 优化思路: 获取当前页数据所有的用户id,做一次in查询,拿到结果后对应userId设置name
            WmUser wmUser = userService.getById(wmNews.getUserId());
            vo.setAuthorName(wmUser.getName());
            // 加入到集合中
            vos.add(vo);
        }
        // 返回通用的分页响应
        PageResponseResult result = new PageResponseResult(dto.getPage(), dto.getSize(), iPage.getTotal(), vos);
        return result;
    }

    @Override
    public ResponseResult<WmNewsVo> findNewVoById(Integer id) {
        // 根据id查询文章
        WmNews wmNews = this.getById(id);
        // 查询作者名称
        WmUser wmUser = userService.getById(wmNews.getUserId());
        // 构建vo对象
        WmNewsVo vo = new WmNewsVo();
        BeanUtils.copyProperties(wmNews, vo);
        vo.setAuthorName(wmUser.getName());
        // 通用响应
        return ResponseResult.okResult(vo);
    }

    @Override
    public ResponseResult auth(NewsAuthDto dto, int type) {
        // 查询文章
        WmNews wmNews = this.getById(dto.getId());
        // 审核不通过 修改状态为2 记录失败原因
        if (type == 0) {
            wmNews.setStatus(2);
            wmNews.setReason(dto.getMsg());
            this.updateById(wmNews);
        } else {
            // 将状态改为4
            wmNews.setStatus(4);
            this.updateById(wmNews);
            // 审核通过 依旧走自动审核的逻辑
            auditService.audit(wmNews);
        }
        return ResponseResult.okResult();
    }
}
